#include "gtkatspitextprivate.h"
#include "gtkatspieditabletextprivate.h"
#include "gtkatspivalueprivate.h"
+#include "gtkatspiselectionprivate.h"
#include "a11y/atspi/atspi-accessible.h"
#include "a11y/atspi/atspi-text.h"
#include "a11y/atspi/atspi-editabletext.h"
#include "a11y/atspi/atspi-value.h"
+#include "a11y/atspi/atspi-selection.h"
#include "gtkdebug.h"
#include "gtkeditable.h"
self->n_registered_objects++;
}
+ vtable = gtk_atspi_get_selection_vtable (widget);
+ if (vtable)
+ {
+ g_variant_builder_add (&interfaces, "s", "org.a11y.atspi.Selection");
+ self->registration_ids[self->n_registered_objects] =
+ g_dbus_connection_register_object (self->connection,
+ self->context_path,
+ (GDBusInterfaceInfo *) &atspi_selection_interface,
+ vtable,
+ self,
+ NULL,
+ NULL);
+ self->n_registered_objects++;
+ }
+
self->interfaces = g_variant_ref_sink (g_variant_builder_end (&interfaces));
}
}
static void
-emit_selection_changed (GtkAtSpiContext *self,
- const char *kind,
- int cursor_position)
+emit_text_selection_changed (GtkAtSpiContext *self,
+ const char *kind,
+ int cursor_position)
{
g_dbus_connection_emit_signal (self->connection,
NULL,
NULL);
}
+static void
+emit_selection_changed (GtkAtSpiContext *self,
+ const char *kind)
+{
+ g_dbus_connection_emit_signal (self->connection,
+ NULL,
+ self->context_path,
+ "org.a11y.atspi.Event.Object",
+ "SelectionChanged",
+ g_variant_new ("(siiva{sv})",
+ "", 0, 0, g_variant_new_string (""), NULL),
+ NULL);
+}
+
static void
emit_state_changed (GtkAtSpiContext *self,
const char *name,
gtk_at_spi_context_unregister_object (self);
gtk_atspi_disconnect_text_signals (GTK_WIDGET (accessible));
+ gtk_atspi_disconnect_selection_signals (GTK_WIDGET (accessible));
G_OBJECT_CLASS (gtk_at_spi_context_parent_class)->dispose (gobject);
}
GtkAccessible *accessible = gtk_at_context_get_accessible (GTK_AT_CONTEXT (self));
gtk_atspi_connect_text_signals (GTK_WIDGET (accessible),
emit_text_changed,
- emit_selection_changed,
+ emit_text_selection_changed,
self);
+ gtk_atspi_connect_selection_signals (GTK_WIDGET (accessible),
+ emit_selection_changed,
+ self);
gtk_at_spi_context_register_object (self);
G_OBJECT_CLASS (gtk_at_spi_context_parent_class)->constructed (gobject);
return self->context_path;
}
+
+/*< private >
+ * gtk_at_spi_context_to_ref:
+ * @self: a #GtkAtSpiContext
+ *
+ * Returns an ATSPI object reference for the #GtkAtSpiContext.
+ *
+ * Returns: (transfer floating): a #GVariant with the reference
+ */
+GVariant *
+gtk_at_spi_context_to_ref (GtkAtSpiContext *self)
+{
+ const char *name = g_dbus_connection_get_unique_name (self->connection);
+ return g_variant_new ("(so)", name, self->context_path);
+}
--- /dev/null
+/* gtkatspiselection.c: AT-SPI Selection implementation
+ *
+ * Copyright 2020 Red Hat, Inc.
+ *
+ * SPDX-License-Identifier: LGPL-2.1-or-later
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2.1 of the License, or (at your option) any later version.
+ *
+ * This library is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public
+ * License along with this library; if not, see <http://www.gnu.org/licenses/>.
+ */
+
+#include "config.h"
+
+#include "gtkatspiselectionprivate.h"
+
+#include "a11y/atspi/atspi-selection.h"
+
+#include "gtkatcontextprivate.h"
+#include "gtkatspicontextprivate.h"
+#include "gtkaccessibleprivate.h"
+#include "gtkdebug.h"
+#include "gtklistbox.h"
+
+#include <gio/gio.h>
+
+typedef struct {
+ int n;
+ GtkListBoxRow *row;
+} RowCounter;
+
+static void
+find_nth (GtkListBox *box,
+ GtkListBoxRow *row,
+ gpointer data)
+{
+ RowCounter *counter = data;
+
+ if (counter->n == 0)
+ counter->row = row;
+
+ counter->n--;
+}
+
+static void
+listbox_handle_method (GDBusConnection *connection,
+ const gchar *sender,
+ const gchar *object_path,
+ const gchar *interface_name,
+ const gchar *method_name,
+ GVariant *parameters,
+ GDBusMethodInvocation *invocation,
+ gpointer user_data)
+{
+ GtkATContext *self = user_data;
+ GtkAccessible *accessible = gtk_at_context_get_accessible (self);
+ GtkWidget *widget = GTK_WIDGET (accessible);
+
+ if (g_strcmp0 (method_name, "GetSelectedChild") == 0)
+ {
+ RowCounter counter;
+ int idx;
+
+ g_variant_get (parameters, "(i)", &idx);
+
+ counter.n = idx;
+ counter.row = NULL;
+
+ gtk_list_box_selected_foreach (GTK_LIST_BOX (widget), find_nth, &counter);
+
+ if (counter.row == NULL)
+ g_dbus_method_invocation_return_error (invocation, G_DBUS_ERROR, G_DBUS_ERROR_INVALID_ARGS, "No selected child for %d", idx);
+ else
+ {
+ GtkATContext *ctx = gtk_accessible_get_at_context (GTK_ACCESSIBLE (counter.row));
+ g_dbus_method_invocation_return_value (invocation, g_variant_new ("(@(so))", gtk_at_spi_context_to_ref (GTK_AT_SPI_CONTEXT (ctx))));
+ }
+ }
+ else if (g_strcmp0 (method_name, "SelectChild") == 0)
+ {
+ int idx;
+ GtkListBoxRow *row;
+
+ g_variant_get (parameters, "(i)", &idx);
+
+ row = gtk_list_box_get_row_at_index (GTK_LIST_BOX (widget), idx);
+ if (!row)
+ g_dbus_method_invocation_return_error (invocation, G_DBUS_ERROR, G_DBUS_ERROR_INVALID_ARGS, "No child at position %d", idx);
+ else
+ {
+ gboolean ret;
+
+ gtk_list_box_select_row (GTK_LIST_BOX (widget), row);
+ ret = gtk_list_box_row_is_selected (row);
+ g_dbus_method_invocation_return_value (invocation, g_variant_new ("(b)", ret));
+ }
+ }
+ else if (g_strcmp0 (method_name, "DeselectChild") == 0)
+ {
+ int idx;
+ GtkListBoxRow *row;
+
+ g_variant_get (parameters, "(i)", &idx);
+
+ row = gtk_list_box_get_row_at_index (GTK_LIST_BOX (widget), idx);
+ if (!row)
+ g_dbus_method_invocation_return_error (invocation, G_DBUS_ERROR, G_DBUS_ERROR_INVALID_ARGS, "No child at position %d", idx);
+ else
+ {
+ gboolean ret;
+
+ gtk_list_box_unselect_row (GTK_LIST_BOX (widget), row);
+ ret = !gtk_list_box_row_is_selected (row);
+ g_dbus_method_invocation_return_value (invocation, g_variant_new ("(b)", ret));
+ }
+ }
+ else if (g_strcmp0 (method_name, "DeselectSelectedChild") == 0)
+ {
+ RowCounter counter;
+ int idx;
+
+ g_variant_get (parameters, "(i)", &idx);
+
+ counter.n = idx;
+ counter.row = NULL;
+
+ gtk_list_box_selected_foreach (GTK_LIST_BOX (widget), find_nth, &counter);
+
+ if (counter.row == NULL)
+ g_dbus_method_invocation_return_error (invocation, G_DBUS_ERROR, G_DBUS_ERROR_INVALID_ARGS, "No selected child for %d", idx);
+ else
+ {
+ gboolean ret;
+
+ gtk_list_box_unselect_row (GTK_LIST_BOX (widget), counter.row);
+ ret = !gtk_list_box_row_is_selected (counter.row);
+ g_dbus_method_invocation_return_value (invocation, g_variant_new ("(b)", ret));
+ }
+ }
+ else if (g_strcmp0 (method_name, "IsChildSelected") == 0)
+ {
+ int idx;
+ GtkListBoxRow *row;
+
+ g_variant_get (parameters, "(i)", &idx);
+
+ row = gtk_list_box_get_row_at_index (GTK_LIST_BOX (widget), idx);
+ if (!row)
+ g_dbus_method_invocation_return_error (invocation, G_DBUS_ERROR, G_DBUS_ERROR_INVALID_ARGS, "No child at position %d", idx);
+ else
+ g_dbus_method_invocation_return_value (invocation, g_variant_new ("(b)", gtk_list_box_row_is_selected (row)));
+ }
+ else if (g_strcmp0 (method_name, "SelectAll") == 0)
+ {
+ gtk_list_box_select_all (GTK_LIST_BOX (widget));
+ g_dbus_method_invocation_return_value (invocation, g_variant_new ("(b)", TRUE));
+ }
+ else if (g_strcmp0 (method_name, "ClearSelection") == 0)
+ {
+ gtk_list_box_unselect_all (GTK_LIST_BOX (widget));
+ g_dbus_method_invocation_return_value (invocation, g_variant_new ("(b)", TRUE));
+ }
+}
+
+static void
+count_selected (GtkListBox *box,
+ GtkListBoxRow *row,
+ gpointer data)
+{
+ *(int *)data += 1;
+}
+
+static GVariant *
+listbox_get_property (GDBusConnection *connection,
+ const gchar *sender,
+ const gchar *object_path,
+ const gchar *interface_name,
+ const gchar *property_name,
+ GError **error,
+ gpointer user_data)
+{
+ GtkATContext *self = GTK_AT_CONTEXT (user_data);
+ GtkAccessible *accessible = gtk_at_context_get_accessible (self);
+ GtkWidget *widget = GTK_WIDGET (accessible);
+
+ if (g_strcmp0 (property_name, "NSelectedChildren") == 0)
+ {
+ int count = 0;
+
+ gtk_list_box_selected_foreach (GTK_LIST_BOX (widget), count_selected, &count);
+
+ return g_variant_new_int32 (count);
+ }
+
+ return NULL;
+}
+
+static const GDBusInterfaceVTable listbox_vtable = {
+ listbox_handle_method,
+ listbox_get_property,
+ NULL
+};
+
+const GDBusInterfaceVTable *
+gtk_atspi_get_selection_vtable (GtkWidget *widget)
+{
+ if (GTK_IS_LIST_BOX(widget))
+ return &listbox_vtable;
+
+ return NULL;
+}
+
+typedef struct {
+ GtkAtspiSelectionCallback *changed;
+ gpointer data;
+} SelectionChanged;
+
+void
+gtk_atspi_connect_selection_signals (GtkWidget *widget,
+ GtkAtspiSelectionCallback selection_changed,
+ gpointer data)
+{
+ if (GTK_IS_LIST_BOX (widget))
+ {
+ SelectionChanged *changed;
+
+ changed = g_new (SelectionChanged, 1);
+ changed->changed = selection_changed;
+ changed->data = data;
+
+ g_object_set_data_full (G_OBJECT (widget), "accessible-selection-data", changed, g_free);
+
+ g_signal_connect_swapped (widget, "selected-rows-changed", G_CALLBACK (selection_changed), data);
+ }
+}
+
+void
+gtk_atspi_disconnect_selection_signals (GtkWidget *widget)
+{
+ if (GTK_IS_LIST_BOX (widget))
+ {
+ SelectionChanged *changed;
+
+ changed = g_object_get_data (G_OBJECT (widget), "accessible-selection-data");
+
+ g_signal_handlers_disconnect_by_func (widget, changed->changed, changed->data);
+
+ g_object_set_data (G_OBJECT (widget), "accessible-selection-data", NULL);
+ }
+}
+